Reiniciar R

Estructuración de Bases de Datos

A lo largo de esta clase, trabajaremos con el paquete Tidyverse. El mismo agrupa una serie de paquetes que tienen una misma lógica en su diseño y por ende funcionan en armonía.
Entre ellos usaremos principalmente dplyr y tidyr para realizar transformaciones sobre nuestro set de datos, y ggplot para realizar gráficos (éste último se verá en la clase 3).

A continuación cargamos la librería a nuestro ambiente. Para ello debe estar previamente instalada en nuestra pc.

library(tidyverse)
library(openxlsx)

Carga de Informacion

La función list.files nos permite observar los archivos que contiene una determinada carpeta

list.files("../Fuentes/")
 [1] "ADEQUI.xlsx"                             
 [2] "Aglomerados EPH.xlsx"                    
 [3] "CANASTAS.xlsx"                           
 [4] "codigo_aglo.xlsx"                        
 [5] "Cuestionario Individual_EPH_Continua.pdf"
 [6] "EPH_Conceptos_Actividad.pdf"             
 [7] "EPH_cont_1trim17.pdf"                    
 [8] "EPH_registro_2_trim_2016.pdf"            
 [9] "informe_EPH_pobreza_02_16.pdf"           
[10] "Metodologia_EPHContinua 2003.pdf"        
[11] "Regiones.xlsx"                           
[12] "usu_hogar_t117.txt"                      
[13] "usu_individual_t117.txt"                 
[14] "usu_individual_t216.txt"                 
[15] "usu_individual_t316.txt"                 
[16] "usu_individual_t416.txt"                 

La función read.table nos permite levantar los archivos de extensión “.txt”
La función read.xlsx nos permite levantar los archivos de extensión “.xlsx”

Levantamos la base individual del primer trimestre de 2017, y un listado que contiene los Nombres y Códigos de los Aglomerados EPH.

Individual_t117 <-
  read.table("../Fuentes/usu_individual_t117.txt",
  sep = ";",
  dec = ",",
  header = TRUE,
  fill = TRUE )
  
  
Aglom <- read.xlsx("../Fuentes/Aglomerados EPH.xlsx")

Para mostrar el funcionamiento básico de tydyverse, crearemos un pequeño subset de datos de la base Individual de la EPH. A partir del lenguaje base, especificaremos conservar las primeras 5000 filas, y una serie de variables de interés, tales como los Códigos identificadorios de las vivindas, hogares y componentes, así como las variables de sexo, edad, y la ponderación.

Datos  <- Individual_t117[1:5000,
                          c("CODUSU","NRO_HOGAR",
                                  "COMPONENTE","AGLOMERADO","CH04","CH06","PONDERA")]

Téngase en cuenta que los resultados de estos ejercicios no serán representativos dado que se trata de un recorte de la base.

Dplyr

El caracter principal para utilizar este paquete es %>% , pipe (de tubería).

Los %>% toman el set de datos a su izquierda, y los transforman mediante los comandos a su derecha, en los cuales los elementos de la izquierda están implícitos. Es decír, que una vez específicado el DataFrame con el cual se trabaja, no será necesario nombrarlo nuevamente para referirse a una determinada variable/columna del mismo.

Veamos las principales funciones que pueden utilizarse con la lógica de este paquete:

filter

Permite filtrar la tabla acorde al cumplimiento de condiciones lógicas

Datos %>% 
  filter(CH04==1 , CH06>=50)

Nótese que en este caso al separar con una , las condiciones se exige el cumplimiento de ambas. En caso de desear que se cumpla al menos una condición debe utilizarse el caracter |

Datos %>% 
    filter(CH04==1| CH06>=50)

rename

Permite renombrar una columna de la tabla. Funciona de la siguiente manera: Data %>% rename( nuevo_nombre = viejo_nombre )

Datos <- Datos %>% 
  rename(EDAD = CH06)
Datos

Nótese que a diferencia del ejemplo de la función filter donde utilizábamos == para comprobar una condición lógica, en este caso se utiliza sólo un = ya que lo estamos haciendo es asignar un nombre.

mutate

Permite agregar una variable a la tabla (especificando el nombre que tomará esta), que puede ser el resultado de operaciones sobre otras variables de la misma tabla.

En caso de especificar el nombre de una columna existente, el resultado de la operación realizada “sobrescribirá” la información de la columna con dicho nombre

Datos <- Datos %>% 
  mutate(Edad_Cuadrado=EDAD^2)
Datos

case_when

Permite definir una variable, la cual toma un valor particular para cada condición establecida. En caso de no cumplir ninguna de las condiciones establecidas la variable tomara valor NA.
Su funcionamiento es el siguiente:
case_when(condicion1 ~ "Valor1",condicion2 ~ "Valor2",condicion3 ~ "Valor3")

Datos <- Datos %>% 
  mutate(Grupos_Etarios = case_when(EDAD  < 18   ~ "Menores",
                                 EDAD  %in%  18:65   ~ "Adultos",
                                 EDAD  > 65   ~ "Adultos Mayores"))
Datos

select

Permite especificar la serie de columnas que se desea conservar de un DataFrame. También pueden especificarse las columnas que se desean descartar (agregándoles un -). Muy útil para agilizar el trabajo en bases de datos de gran tamaño.

Datos %>% 
  select(CODUSU,PONDERA)
Datos %>% 
  select(-c(PONDERA,EDAD))

arrange

Permite ordenar la tabla por los valores de determinada/s variable/s. Es útil cuando luego deben hacerse otras operaciones que requieran del ordenamiento de la tabla

Datos <- Datos %>% 
  arrange(CH04,EDAD)
Datos

summarise

Crea una nueva tabla que resume la información original. Para ello, definimos las variables de resumen y las formas de agregación.

Datos %>% 
  summarise(Edad_prom = mean(EDAD),
            Edad_prom_pond = weighted.mean(EDAD,PONDERA))

group_by

Esta función permite realizar operaciones de forma agrupada. Lo que hace la función es “separar” a la tabla según los valores de la variable indicada y realizar las operaciones que se especifican a continuación, de manera independiente para cada una de las “subtablas”. En nuestro ejemplo, sería útil para calcular el promedio de edad según sexo

Datos %>% 
  group_by(CH04) %>%
  summarise(Edad_Prom = weighted.mean(EDAD,PONDERA))

Notese que los %>% pueden usarse encadenados para realizar numerosos procedimientos sobre un dataframe original. Veamos un ejemplo con multiples encadenamietnos

Encadenado <- Datos %>% 
  filter(Grupos_Etarios == "Adultos") %>% 
  rename(Codigo = CODUSU) %>% 
  mutate(Sexo = case_when(CH04 == 1 ~ "Varon",
                          CH04 == 2 ~ "Mujer")) %>% 
  select(-Edad_Cuadrado)
  
Encadenado

Joins

Otra implementación muy importante del paquete dplyr son las funciones para unir tablas (joins)

left_join

Veamos un ejemplo de la función left_join (una de las más utilizadas en la práctica).
Para ello utilizaremos el Dataframe Aglom con los códigos y los nombres de los aglomerados EPH

Aglom
Datos_join <- Datos %>% 
  left_join(.,Aglom, by = "AGLOMERADO")
Datos_join
Poblacion_Aglomerados <- Datos_join %>% 
  filter(AGLOMERADO != 5) %>% 
  group_by(Nom_Aglo) %>% 
  summarise(Varones = sum(PONDERA[CH04==1]),
            Mujeres = sum(PONDERA[CH04==2]))
Poblacion_Aglomerados

Tidyr

El paquete tidyr esta pensado para facilitar el emprolijamiento de los datos.

Gather es una función que nos permite pasar los datos de forma horizontal a una forma vertical.

spread es una función que nos permite pasar los datos de forma vertical a una forma horizontal.

Gather y Spread

La función gather nos permite transformar múltiples variables hacia una sola, generando una columna que identificará como categoría lo que antes era una variable. Este tipo de procesamientos será fundamental a la hora de realizar gráficos.

Datos_gather <- Poblacion_Aglomerados %>%  
  gather(.,         # el . llama a lo que esta atras del %>% 
   key   = Grupo,   # como se llamará la variable que toma los nombres de las columnas 
   value = Total,  # como se llamará la variable que toma los valores de las columnas
   2:3)             #le indico que columnas juntar
Datos_gather

La función spread realiza el procedimiento inverso, y puede ser util para presentar los datos en forma de tabla de manera más prolija.

Datos_Spread <- Datos_gather %>% 
  spread(.,       # el . llama a lo que esta atras del %>% 
  key = Grupo,    #la llave es la variable cuyos valores van a dar los nombres de columnas
  value = Total) #los valores con que se llenan las celdas
Datos_Spread  

Tasas del Mercado de Trabajo

Luego de abordar las principales funciones necesarias para operar sobre las bases de datos, trabajaremos a continuación con toda la base del 1er Trimestre de 2017 de la EPH. El ejercicio principal consistirá en calcular la tasa de empleo, definida como:

  • Tasa de empleo: \(\frac{Ocupados}{Población}\)

En la carpeta de FUENTES del curso, se encuentra el archivo “EPH_Registro” que contiene las codificación de cada una de las variables de la base, y el archivo “EPH_Concpetos_Actividad” que contiene las definiciones de los Estados ocupacionales a partir de los cuales se construyen las tasas básicas.

Tasa de Empleo

Creamos una tabla con los niveles de:

  • Población
  • Ocupados

Estos niveles nos van a permitir calcular la tasa de forma sencilla.

  • Población: Si contaramos cuantos registros tiene la base, simplemente tendríamos el numero de individuos muestral de la EPH, por ende debemos sumar los valores de la variable PONDERA, para contemplar a cuantas personas representa cada individuo encuestado.
  • Ocupados: En este caso, debemos agregar un filtro al procedimiento anterior, ya que unicamente queremos sumar los ponderadores de aquellas personas que se encuentran ocupadas. (La lógica seria: “Suma los valores de la columna PONDERA, solo para aquellos registros donde el ESTADO == 1”)
Poblacion_ocupados <- Individual_t117 %>% 
  summarise(Poblacion         = sum(PONDERA),
            Ocupados          = sum(PONDERA[ESTADO == 1]))
Poblacion_ocupados

La función summarise() nos permite crear multiples variables de resumen al mismo tiempo, simplemente separando con una , cada uno de ellas. A su vez, se pueden crear variables, a partir de las variables creadas por la propia función. De esta forma, podemos, directamente calcular la tasa de empleo a partir del total poblacional y de ocupados.

Empleo <- Individual_t117 %>% 
  summarise(Poblacion         = sum(PONDERA),
            Ocupados          = sum(PONDERA[ESTADO == 1]),
            Tasa_Empleo    = Ocupados/Poblacion)
Empleo

En caso de querer expresar los resultados como porcentajes, utilizamos la función sprintf. Para ello debemos utilizar mutate para transformar la columna Tasa_Empleo

Empleo %>% 
  mutate(Tasa_Empleo_Porc = sprintf("%1.1f%%", 100*Tasa_Empleo))

Nótese que en este caso, para poder añadir el %, la función transforma a la variable en un Character, por ende debe tenerse en cuenta que se pierde la información del numero completo.

Exportar resultados a Excel

Como vieramos en la clase 1, la función write.xlsx de la libreria openxlsx nos permite exportar los dos dataframes a un mismo archivo. Cabe aclarar que existen numerosas funciones y librerías alternativas para exportar resultados a un excel. En este caso, optamos por openxlsx ya que resulta una de las más sencillas para exportar rapidamente los resultados. Otras librerías permiten también dar formato a las tablas que se exportan, definir si deseamos sobreescribir archivos en caso de que ya existan, etc.

Lista_a_exportar <- list("Tasa de Empleo" = Empleo,
                         "Poblacion Aglos" = Poblacion_Aglomerados)

write.xlsx(Lista_a_exportar,"../Resultados/Informe Mercado de Trabajo.xlsx")
LS0tDQp0aXRsZTogIkludHJvZHVjY2nDs24gYSBSIHBhcmEgQ2llbmNpYXMgU29jaWFsZXMuIEFwbGljYWNpw7NuIHByw6FjdGljYSBlbiBsYSBFUEgiDQpzdWJ0aXRsZTogIkNsYXNlIDMgLSBUaWR5dmVyc2UgeSBUYXNhcyBiw6FzaWNhcyBkZWwgTWVyY2FkbyBkZSBUcmFiYWpvIg0KZGF0ZTogIjIwLzA4LzIwMTkiDQpvdXRwdXQ6DQogIGh0bWxfbm90ZWJvb2s6DQogICAgdG9jOiB5ZXMNCiAgICB0b2NfZmxvYXQ6IHllcw0KICBodG1sX2RvY3VtZW50Og0KICAgIHRvYzogeWVzDQotLS0NCg0KPiBSZWluaWNpYXIgUg0KDQojIEVzdHJ1Y3R1cmFjacOzbiBkZSBCYXNlcyBkZSBEYXRvcw0KICAgICAgICAgDQoNCkEgbG8gbGFyZ28gZGUgZXN0YSBjbGFzZSwgdHJhYmFqYXJlbW9zIGNvbiBlbCBwYXF1ZXRlIFtUaWR5dmVyc2VdKGh0dHBzOi8vd3d3LnRpZHl2ZXJzZS5vcmcvKS4gRWwgbWlzbW8gYWdydXBhIHVuYSBzZXJpZSBkZSBwYXF1ZXRlcyBxdWUgdGllbmVuIHVuYSBtaXNtYSBsw7NnaWNhIGVuIHN1IGRpc2XDsW8geSBwb3IgZW5kZSBmdW5jaW9uYW4gZW4gYXJtb27DrWEuICAgICANCkVudHJlIGVsbG9zIHVzYXJlbW9zIHByaW5jaXBhbG1lbnRlIF9fZHBseXJfXyB5IF9fdGlkeXJfXyBwYXJhIHJlYWxpemFyIHRyYW5zZm9ybWFjaW9uZXMgc29icmUgbnVlc3RybyBzZXQgZGUgZGF0b3MsIHkgX19nZ3Bsb3RfXyBwYXJhIHJlYWxpemFyIGdyw6FmaWNvcyAow6lzdGUgw7psdGltbyBzZSB2ZXLDoSBlbiBsYSBjbGFzZSAzKS4NCg0KQSBjb250aW51YWNpw7NuIGNhcmdhbW9zIGxhIGxpYnJlcsOtYSBhIG51ZXN0cm8gYW1iaWVudGUuIFBhcmEgZWxsbyBkZWJlIGVzdGFyIHByZXZpYW1lbnRlIGluc3RhbGFkYSBlbiBudWVzdHJhIHBjLg0KYGBge3IsIHdhcm5pbmc9RkFMU0UsbWVzc2FnZT1GQUxTRX0NCmxpYnJhcnkodGlkeXZlcnNlKQ0KbGlicmFyeShvcGVueGxzeCkNCmBgYA0KDQojIyBDYXJnYSBkZSBJbmZvcm1hY2lvbg0KDQpMYSBmdW5jacOzbiBfX2xpc3QuZmlsZXNfXyBub3MgcGVybWl0ZSBvYnNlcnZhciBsb3MgYXJjaGl2b3MgcXVlIGNvbnRpZW5lIHVuYSBkZXRlcm1pbmFkYSBjYXJwZXRhICAgICAgICAgICAgIA0KDQpgYGB7cn0NCmxpc3QuZmlsZXMoIi4uL0Z1ZW50ZXMvIikNCmBgYA0KTGEgZnVuY2nDs24gX19yZWFkLnRhYmxlX18gbm9zIHBlcm1pdGUgbGV2YW50YXIgbG9zIGFyY2hpdm9zIGRlIGV4dGVuc2nDs24gIi50eHQiICAgICAgICAgICAgICAgDQpMYSBmdW5jacOzbiBfX3JlYWQueGxzeF9fIG5vcyBwZXJtaXRlIGxldmFudGFyIGxvcyBhcmNoaXZvcyBkZSBleHRlbnNpw7NuICIueGxzeCIgICAgICAgICAgICAgICAgIA0KDQpMZXZhbnRhbW9zIGxhIGJhc2UgaW5kaXZpZHVhbCBkZWwgcHJpbWVyIHRyaW1lc3RyZSBkZSAyMDE3LCB5IHVuIGxpc3RhZG8gcXVlIGNvbnRpZW5lIGxvcyBOb21icmVzIHkgQ8OzZGlnb3MgZGUgbG9zIEFnbG9tZXJhZG9zIEVQSC4NCmBgYHtyfQ0KSW5kaXZpZHVhbF90MTE3IDwtDQogIHJlYWQudGFibGUoIi4uL0Z1ZW50ZXMvdXN1X2luZGl2aWR1YWxfdDExNy50eHQiLA0KICBzZXAgPSAiOyIsDQogIGRlYyA9ICIsIiwNCiAgaGVhZGVyID0gVFJVRSwNCiAgZmlsbCA9IFRSVUUgKQ0KICANCiAgDQpBZ2xvbSA8LSByZWFkLnhsc3goIi4uL0Z1ZW50ZXMvQWdsb21lcmFkb3MgRVBILnhsc3giKQ0KYGBgDQoNCg0KUGFyYSBtb3N0cmFyIGVsIGZ1bmNpb25hbWllbnRvIGLDoXNpY28gZGUgdHlkeXZlcnNlLCBjcmVhcmVtb3MgdW4gcGVxdWXDsW8gc3Vic2V0IGRlIGRhdG9zIGRlIGxhIGJhc2UgSW5kaXZpZHVhbCBkZSBsYSBFUEguIEEgcGFydGlyIGRlbCBsZW5ndWFqZSBiYXNlLCBlc3BlY2lmaWNhcmVtb3MgY29uc2VydmFyIGxhcyBwcmltZXJhcyA1MDAwIGZpbGFzLCB5IHVuYSBzZXJpZSBkZSB2YXJpYWJsZXMgZGUgaW50ZXLDqXMsIHRhbGVzIGNvbW8gbG9zIEPDs2RpZ29zIGlkZW50aWZpY2Fkb3Jpb3MgZGUgbGFzIHZpdmluZGFzLCBob2dhcmVzIHkgY29tcG9uZW50ZXMsIGFzw60gY29tbyBsYXMgdmFyaWFibGVzIGRlIHNleG8sIGVkYWQsIHkgbGEgcG9uZGVyYWNpw7NuLg0KYGBge3J9DQpEYXRvcyAgPC0gSW5kaXZpZHVhbF90MTE3WzE6NTAwMCwNCiAgICAgICAgICAgICAgICAgICAgICAgICAgYygiQ09EVVNVIiwiTlJPX0hPR0FSIiwNCiAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAiQ09NUE9ORU5URSIsIkFHTE9NRVJBRE8iLCJDSDA0IiwiQ0gwNiIsIlBPTkRFUkEiKV0NCg0KYGBgDQpUw6luZ2FzZSBlbiBjdWVudGEgcXVlIGxvcyByZXN1bHRhZG9zIGRlIGVzdG9zIGVqZXJjaWNpb3Mgbm8gc2Vyw6FuIHJlcHJlc2VudGF0aXZvcyBkYWRvIHF1ZSBzZSB0cmF0YSBkZSB1biByZWNvcnRlIGRlIGxhIGJhc2UuDQoNCiMjIERwbHlyDQoNCkVsIGNhcmFjdGVyIHByaW5jaXBhbCBwYXJhIHV0aWxpemFyIGVzdGUgcGFxdWV0ZSBlcyBgYGAlPiVgYGAgLCBfcGlwZV8gKGRlIHR1YmVyw61hKS4NCg0KTG9zIGBgYCU+JWBgYCB0b21hbiBlbCBzZXQgZGUgZGF0b3MgYSBzdSBpenF1aWVyZGEsIHkgbG9zIHRyYW5zZm9ybWFuIG1lZGlhbnRlIGxvcyBjb21hbmRvcyBhIHN1IGRlcmVjaGEsIGVuIGxvcyBjdWFsZXMgbG9zIGVsZW1lbnRvcyBkZSBsYSBpenF1aWVyZGEgZXN0w6FuIGltcGzDrWNpdG9zLiBFcyBkZWPDrXIsIHF1ZSB1bmEgdmV6IGVzcGVjw61maWNhZG8gZWwgRGF0YUZyYW1lIGNvbiBlbCBjdWFsIHNlIHRyYWJhamEsIG5vIHNlcsOhIG5lY2VzYXJpbyBub21icmFybG8gbnVldmFtZW50ZSBwYXJhIHJlZmVyaXJzZSBhIHVuYSBkZXRlcm1pbmFkYSB2YXJpYWJsZS9jb2x1bW5hIGRlbCBtaXNtby4NCg0KVmVhbW9zIGxhcyBwcmluY2lwYWxlcyBmdW5jaW9uZXMgcXVlIHB1ZWRlbiB1dGlsaXphcnNlIGNvbiBsYSBsw7NnaWNhIGRlIGVzdGUgcGFxdWV0ZToNCg0KIyMjIGZpbHRlcg0KDQpQZXJtaXRlIGZpbHRyYXIgbGEgdGFibGEgYWNvcmRlIGFsIGN1bXBsaW1pZW50byBkZSBjb25kaWNpb25lcyBsw7NnaWNhcw0KIA0KYGBge3J9DQpEYXRvcyAlPiUgDQogIGZpbHRlcihDSDA0PT0xICwgQ0gwNj49NTApDQoNCmBgYA0KTsOzdGVzZSBxdWUgZW4gZXN0ZSBjYXNvIGFsIHNlcGFyYXIgY29uIHVuYSAgYGBgLGBgYCBsYXMgY29uZGljaW9uZXMgc2UgZXhpZ2UgZWwgY3VtcGxpbWllbnRvIGRlIGFtYmFzLiBFbiBjYXNvIGRlIGRlc2VhciBxdWUgc2UgY3VtcGxhIGFsIG1lbm9zIHVuYSAgY29uZGljacOzbiBkZWJlIHV0aWxpemFyc2UgZWwgY2FyYWN0ZXIgYGBgfGBgYA0KYGBge3J9DQpEYXRvcyAlPiUgDQogICAgZmlsdGVyKENIMDQ9PTF8IENIMDY+PTUwKQ0KDQpgYGANCg0KIyMjIHJlbmFtZQ0KUGVybWl0ZSByZW5vbWJyYXIgdW5hIGNvbHVtbmEgZGUgbGEgdGFibGEuIEZ1bmNpb25hIGRlIGxhIHNpZ3VpZW50ZSBtYW5lcmE6IA0KIGBgYERhdGEgJT4lIHJlbmFtZSggbnVldm9fbm9tYnJlID0gdmllam9fbm9tYnJlIClgYGAgDQpgYGB7cn0NCkRhdG9zIDwtIERhdG9zICU+JSANCiAgcmVuYW1lKEVEQUQgPSBDSDA2KQ0KRGF0b3MNCmBgYA0KTsOzdGVzZSBxdWUgYSBkaWZlcmVuY2lhIGRlbCBlamVtcGxvIGRlIGxhIGZ1bmNpw7NuIF9fZmlsdGVyX18gZG9uZGUgdXRpbGl6w6FiYW1vcyBfXz09X18gcGFyYSBjb21wcm9iYXIgdW5hIGNvbmRpY2nDs24gbMOzZ2ljYSwgZW4gZXN0ZSBjYXNvIHNlIHV0aWxpemEgc8OzbG8gdW4gX189X18geWEgcXVlIGxvIGVzdGFtb3MgaGFjaWVuZG8gZXMgX2FzaWduYXJfIHVuIG5vbWJyZS4NCg0KIyMjIG11dGF0ZQ0KUGVybWl0ZSBhZ3JlZ2FyIHVuYSB2YXJpYWJsZSBhIGxhIHRhYmxhIChlc3BlY2lmaWNhbmRvIGVsIG5vbWJyZSBxdWUgdG9tYXLDoSBlc3RhKSwgcXVlIHB1ZWRlIHNlciBlbCByZXN1bHRhZG8gZGUgb3BlcmFjaW9uZXMgc29icmUgb3RyYXMgdmFyaWFibGVzIGRlIGxhIG1pc21hIHRhYmxhLiAgICAgICANCg0KRW4gY2FzbyBkZSBlc3BlY2lmaWNhciBlbCBub21icmUgZGUgdW5hIGNvbHVtbmEgZXhpc3RlbnRlLCBlbCByZXN1bHRhZG8gZGUgbGEgb3BlcmFjacOzbiByZWFsaXphZGEgInNvYnJlc2NyaWJpcsOhIiBsYSBpbmZvcm1hY2nDs24gZGUgbGEgY29sdW1uYSBjb24gZGljaG8gbm9tYnJlDQpgYGB7cn0NCkRhdG9zIDwtIERhdG9zICU+JSANCiAgbXV0YXRlKEVkYWRfQ3VhZHJhZG89RURBRF4yKQ0KRGF0b3MNCmBgYA0KDQojIyMgY2FzZV93aGVuDQpQZXJtaXRlIGRlZmluaXIgdW5hIHZhcmlhYmxlLCBsYSBjdWFsIHRvbWEgdW4gdmFsb3IgcGFydGljdWxhciBwYXJhIGNhZGEgY29uZGljacOzbiBlc3RhYmxlY2lkYS4gRW4gY2FzbyBkZSBubyBjdW1wbGlyIG5pbmd1bmEgZGUgbGFzIGNvbmRpY2lvbmVzIGVzdGFibGVjaWRhcyBsYSB2YXJpYWJsZSB0b21hcmEgdmFsb3IgX19OQV9fLiAgICAgICAgIA0KU3UgZnVuY2lvbmFtaWVudG8gZXMgZWwgc2lndWllbnRlOiAgICAgIA0KYGBgY2FzZV93aGVuKGNvbmRpY2lvbjEgfiAiVmFsb3IxIixjb25kaWNpb24yIH4gIlZhbG9yMiIsY29uZGljaW9uMyB+ICJWYWxvcjMiKWBgYA0KDQpgYGB7cn0NCkRhdG9zIDwtIERhdG9zICU+JSANCiAgbXV0YXRlKEdydXBvc19FdGFyaW9zID0gY2FzZV93aGVuKEVEQUQgIDwgMTggICB+ICJNZW5vcmVzIiwNCiAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgIEVEQUQgICVpbiUgIDE4OjY1ICAgfiAiQWR1bHRvcyIsDQogICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICBFREFEICA+IDY1ICAgfiAiQWR1bHRvcyBNYXlvcmVzIikpDQpEYXRvcw0KYGBgDQoNCiMjIyBzZWxlY3QNClBlcm1pdGUgZXNwZWNpZmljYXIgbGEgc2VyaWUgZGUgY29sdW1uYXMgcXVlIHNlIGRlc2VhIGNvbnNlcnZhciBkZSB1biBEYXRhRnJhbWUuIFRhbWJpw6luIHB1ZWRlbiBlc3BlY2lmaWNhcnNlIGxhcyBjb2x1bW5hcyBxdWUgc2UgZGVzZWFuIGRlc2NhcnRhciAoYWdyZWfDoW5kb2xlcyB1biBfLV8pLiBNdXkgw7p0aWwgcGFyYSBhZ2lsaXphciBlbCB0cmFiYWpvIGVuIGJhc2VzIGRlIGRhdG9zIGRlIGdyYW4gdGFtYcOxby4NCmBgYHtyfQ0KRGF0b3MgJT4lIA0KICBzZWxlY3QoQ09EVVNVLFBPTkRFUkEpDQoNCg0KRGF0b3MgJT4lIA0KICBzZWxlY3QoLWMoUE9OREVSQSxFREFEKSkNCg0KYGBgDQoNCiMjIyBhcnJhbmdlDQpQZXJtaXRlIG9yZGVuYXIgbGEgdGFibGEgcG9yIGxvcyB2YWxvcmVzIGRlIGRldGVybWluYWRhL3MgdmFyaWFibGUvcy4gRXMgw7p0aWwgY3VhbmRvIGx1ZWdvIGRlYmVuIGhhY2Vyc2Ugb3RyYXMgb3BlcmFjaW9uZXMgcXVlIHJlcXVpZXJhbiBkZWwgb3JkZW5hbWllbnRvIGRlIGxhIHRhYmxhDQpgYGB7cn0NCkRhdG9zIDwtIERhdG9zICU+JSANCiAgYXJyYW5nZShDSDA0LEVEQUQpDQpEYXRvcw0KYGBgDQoNCiMjIyBzdW1tYXJpc2UNCkNyZWEgdW5hIG51ZXZhIHRhYmxhIHF1ZSByZXN1bWUgbGEgaW5mb3JtYWNpw7NuIG9yaWdpbmFsLiBQYXJhIGVsbG8sIGRlZmluaW1vcyBsYXMgdmFyaWFibGVzIGRlIHJlc3VtZW4geSBsYXMgZm9ybWFzIGRlIGFncmVnYWNpw7NuLg0KYGBge3J9DQpEYXRvcyAlPiUgDQogIHN1bW1hcmlzZShFZGFkX3Byb20gPSBtZWFuKEVEQUQpLA0KICAgICAgICAgICAgRWRhZF9wcm9tX3BvbmQgPSB3ZWlnaHRlZC5tZWFuKEVEQUQsUE9OREVSQSkpDQoNCmBgYA0KDQojIyMgZ3JvdXBfYnkNCkVzdGEgZnVuY2nDs24gcGVybWl0ZSByZWFsaXphciBvcGVyYWNpb25lcyBkZSBmb3JtYSBhZ3J1cGFkYS4gTG8gcXVlIGhhY2UgbGEgZnVuY2nDs24gZXMgInNlcGFyYXIiIGEgbGEgdGFibGEgc2Vnw7puIGxvcyB2YWxvcmVzIGRlIGxhIHZhcmlhYmxlIGluZGljYWRhIHkgcmVhbGl6YXIgbGFzIG9wZXJhY2lvbmVzIHF1ZSBzZSBlc3BlY2lmaWNhbiBhICBjb250aW51YWNpw7NuLCBkZSBtYW5lcmEgaW5kZXBlbmRpZW50ZSBwYXJhIGNhZGEgdW5hIGRlIGxhcyAic3VidGFibGFzIi4gRW4gbnVlc3RybyBlamVtcGxvLCBzZXLDrWEgw7p0aWwgcGFyYSBjYWxjdWxhciBlbCBwcm9tZWRpbyBkZSBlZGFkIHNlZ8O6biAgX3NleG9fIA0KYGBge3J9DQpEYXRvcyAlPiUgDQogIGdyb3VwX2J5KENIMDQpICU+JQ0KICBzdW1tYXJpc2UoRWRhZF9Qcm9tID0gd2VpZ2h0ZWQubWVhbihFREFELFBPTkRFUkEpKQ0KYGBgDQpOb3Rlc2UgcXVlIGxvcyBgYGAlPiVgYGAgcHVlZGVuIHVzYXJzZSBlbmNhZGVuYWRvcyBwYXJhIHJlYWxpemFyIG51bWVyb3NvcyBwcm9jZWRpbWllbnRvcyBzb2JyZSB1biBkYXRhZnJhbWUgb3JpZ2luYWwuDQpWZWFtb3MgdW4gZWplbXBsbyBjb24gbXVsdGlwbGVzIGVuY2FkZW5hbWlldG5vcw0KYGBge3J9DQpFbmNhZGVuYWRvIDwtIERhdG9zICU+JSANCiAgZmlsdGVyKEdydXBvc19FdGFyaW9zID09ICJBZHVsdG9zIikgJT4lIA0KICByZW5hbWUoQ29kaWdvID0gQ09EVVNVKSAlPiUgDQogIG11dGF0ZShTZXhvID0gY2FzZV93aGVuKENIMDQgPT0gMSB+ICJWYXJvbiIsDQogICAgICAgICAgICAgICAgICAgICAgICAgIENIMDQgPT0gMiB+ICJNdWplciIpKSAlPiUgDQogIHNlbGVjdCgtRWRhZF9DdWFkcmFkbykNCiAgDQpFbmNhZGVuYWRvDQpgYGANCg0KIyMgSm9pbnMNCg0KT3RyYSBpbXBsZW1lbnRhY2nDs24gbXV5IGltcG9ydGFudGUgZGVsIHBhcXVldGUgZHBseXIgc29uIGxhcyBmdW5jaW9uZXMgcGFyYSB1bmlyIHRhYmxhcyAoam9pbnMpDQoNCg0KIVtmdWVudGU6IGh0dHA6Ly9yc3R1ZGlvLXB1YnMtc3RhdGljLnMzLmFtYXpvbmF3cy5jb20vMjI3MTcxXzYxOGViZGNlMGI5ZDQ0ZjNhZjY1NzAwZTgzMzU5M2RiLmh0bWxdKGpvaW5zLnBuZykgICAgICAgICANCg0KIyMjbGVmdF9qb2luICAgIA0KDQpWZWFtb3MgdW4gZWplbXBsbyBkZSBsYSBmdW5jacOzbiBfX2xlZnRfam9pbl9fICh1bmEgZGUgbGFzIG3DoXMgdXRpbGl6YWRhcyBlbiBsYSBwcsOhY3RpY2EpLiAgICAgICANClBhcmEgZWxsbyB1dGlsaXphcmVtb3MgZWwgRGF0YWZyYW1lIF9BZ2xvbV8gY29uIGxvcyBjw7NkaWdvcyB5IGxvcyBub21icmVzIGRlIGxvcyBhZ2xvbWVyYWRvcyBFUEgNCmBgYHtyfQ0KQWdsb20NCmBgYA0KDQpgYGB7cn0NCkRhdG9zX2pvaW4gPC0gRGF0b3MgJT4lIA0KICBsZWZ0X2pvaW4oLixBZ2xvbSwgYnkgPSAiQUdMT01FUkFETyIpDQpEYXRvc19qb2luDQoNClBvYmxhY2lvbl9BZ2xvbWVyYWRvcyA8LSBEYXRvc19qb2luICU+JSANCiAgZmlsdGVyKEFHTE9NRVJBRE8gIT0gNSkgJT4lIA0KICBncm91cF9ieShOb21fQWdsbykgJT4lIA0KICBzdW1tYXJpc2UoVmFyb25lcyA9IHN1bShQT05ERVJBW0NIMDQ9PTFdKSwNCiAgICAgICAgICAgIE11amVyZXMgPSBzdW0oUE9OREVSQVtDSDA0PT0yXSkpDQoNClBvYmxhY2lvbl9BZ2xvbWVyYWRvcw0KDQpgYGANCg0KIyMgVGlkeXINCg0KRWwgcGFxdWV0ZSB0aWR5ciBlc3RhIHBlbnNhZG8gcGFyYSBmYWNpbGl0YXIgZWwgZW1wcm9saWphbWllbnRvIGRlIGxvcyBkYXRvcy4NCg0KX19HYXRoZXJfXyBlcyB1bmEgZnVuY2nDs24gcXVlIG5vcyBwZXJtaXRlIHBhc2FyIGxvcyBkYXRvcyBkZSBmb3JtYSBob3Jpem9udGFsIGEgdW5hIGZvcm1hIHZlcnRpY2FsLiANCg0KX19zcHJlYWRfXyBlcyB1bmEgZnVuY2nDs24gcXVlIG5vcyBwZXJtaXRlIHBhc2FyIGxvcyBkYXRvcyBkZSBmb3JtYSB2ZXJ0aWNhbCBhIHVuYSBmb3JtYSBob3Jpem9udGFsLg0KDQohW2Z1ZW50ZTogaHR0cDovL3d3dy5naXMtYmxvZy5jb20vZGF0YS1tYW5hZ2VtZW50LXdpdGgtci10aWR5ci1wYXJ0LTEvXShzcHJlYWRWU2dhdGhlci5wbmcpDQoNCiMjIyBHYXRoZXIgeSBTcHJlYWQNCkxhIGZ1bmNpw7NuIGdhdGhlciBub3MgcGVybWl0ZSB0cmFuc2Zvcm1hciBtw7psdGlwbGVzIHZhcmlhYmxlcyBoYWNpYSB1bmEgc29sYSwgZ2VuZXJhbmRvIHVuYSBjb2x1bW5hIHF1ZSBpZGVudGlmaWNhcsOhIGNvbW8gY2F0ZWdvcsOtYSBsbyBxdWUgYW50ZXMgZXJhIHVuYSB2YXJpYWJsZS4gRXN0ZSB0aXBvIGRlIHByb2Nlc2FtaWVudG9zIHNlcsOhIGZ1bmRhbWVudGFsIGEgbGEgaG9yYSBkZSByZWFsaXphciBncsOhZmljb3MuDQoNCmBgYHtyfQ0KDQpEYXRvc19nYXRoZXIgPC0gUG9ibGFjaW9uX0FnbG9tZXJhZG9zICU+JSAgDQogIGdhdGhlciguLCAgICAgICAgICMgZWwgLiBsbGFtYSBhIGxvIHF1ZSBlc3RhIGF0cmFzIGRlbCAlPiUgDQogICBrZXkgICA9IEdydXBvLCAgICMgY29tbyBzZSBsbGFtYXLDoSBsYSB2YXJpYWJsZSBxdWUgdG9tYSBsb3Mgbm9tYnJlcyBkZSBsYXMgY29sdW1uYXMgDQogICB2YWx1ZSA9IFRvdGFsLCAgIyBjb21vIHNlIGxsYW1hcsOhIGxhIHZhcmlhYmxlIHF1ZSB0b21hIGxvcyB2YWxvcmVzIGRlIGxhcyBjb2x1bW5hcw0KICAgMjozKSAgICAgICAgICAgICAjbGUgaW5kaWNvIHF1ZSBjb2x1bW5hcyBqdW50YXINCg0KRGF0b3NfZ2F0aGVyDQpgYGANCkxhIGZ1bmNpw7NuIHNwcmVhZCByZWFsaXphIGVsIHByb2NlZGltaWVudG8gaW52ZXJzbywgeSBwdWVkZSBzZXIgdXRpbCBwYXJhIHByZXNlbnRhciBsb3MgZGF0b3MgZW4gZm9ybWEgZGUgdGFibGEgZGUgbWFuZXJhIG3DoXMgcHJvbGlqYS4gIA0KYGBge3J9DQpEYXRvc19TcHJlYWQgPC0gRGF0b3NfZ2F0aGVyICU+JSANCiAgc3ByZWFkKC4sICAgICAgICMgZWwgLiBsbGFtYSBhIGxvIHF1ZSBlc3RhIGF0cmFzIGRlbCAlPiUgDQogIGtleSA9IEdydXBvLCAgICAjbGEgbGxhdmUgZXMgbGEgdmFyaWFibGUgY3V5b3MgdmFsb3JlcyB2YW4gYSBkYXIgbG9zIG5vbWJyZXMgZGUgY29sdW1uYXMNCiAgdmFsdWUgPSBUb3RhbCkgI2xvcyB2YWxvcmVzIGNvbiBxdWUgc2UgbGxlbmFuIGxhcyBjZWxkYXMNCg0KRGF0b3NfU3ByZWFkICANCmBgYA0KDQoNCiMgVGFzYXMgZGVsIE1lcmNhZG8gZGUgVHJhYmFqbw0KTHVlZ28gZGUgYWJvcmRhciBsYXMgcHJpbmNpcGFsZXMgZnVuY2lvbmVzIG5lY2VzYXJpYXMgcGFyYSBvcGVyYXIgc29icmUgbGFzIGJhc2VzIGRlIGRhdG9zLCB0cmFiYWphcmVtb3MgYSBjb250aW51YWNpw7NuIGNvbiB0b2RhIGxhIGJhc2UgZGVsIDFlciBUcmltZXN0cmUgZGUgMjAxNyBkZSBsYSBFUEguIEVsIGVqZXJjaWNpbyBwcmluY2lwYWwgY29uc2lzdGlyw6EgZW4gY2FsY3VsYXIgbGEgX190YXNhIGRlIGVtcGxlb19fLCBkZWZpbmlkYSBjb21vOiAgICAgICAgICAgIA0KDQotIFRhc2EgZGUgZW1wbGVvOiAgJFxmcmFje09jdXBhZG9zfXtQb2JsYWNpw7NufSQgDQogICAgICANCkVuIGxhIGNhcnBldGEgZGUgX0ZVRU5URVNfIGRlbCBjdXJzbywgc2UgZW5jdWVudHJhIGVsIGFyY2hpdm8gICoiRVBIX1JlZ2lzdHJvIiogcXVlIGNvbnRpZW5lIGxhcyBjb2RpZmljYWNpw7NuIGRlIGNhZGEgdW5hIGRlIGxhcyB2YXJpYWJsZXMgZGUgbGEgYmFzZSwgeSBlbCBhcmNoaXZvICoiRVBIX0NvbmNwZXRvc19BY3RpdmlkYWQiKiBxdWUgY29udGllbmUgbGFzIGRlZmluaWNpb25lcyBkZSBsb3MgRXN0YWRvcyBvY3VwYWNpb25hbGVzIGEgcGFydGlyIGRlIGxvcyBjdWFsZXMgc2UgY29uc3RydXllbiBsYXMgdGFzYXMgYsOhc2ljYXMuICAgDQoNCiMjIFRhc2EgZGUgRW1wbGVvDQoNCkNyZWFtb3MgdW5hIHRhYmxhIGNvbiBsb3Mgbml2ZWxlcyBkZToNCg0KLSBQb2JsYWNpw7NuDQotIE9jdXBhZG9zDQoNCkVzdG9zIG5pdmVsZXMgbm9zIHZhbiBhIHBlcm1pdGlyIGNhbGN1bGFyIGxhIHRhc2EgZGUgZm9ybWEgc2VuY2lsbGEuICAgICAgDQoNCiAtIFBvYmxhY2nDs246IFNpIGNvbnRhcmFtb3MgY3VhbnRvcyByZWdpc3Ryb3MgdGllbmUgbGEgYmFzZSwgc2ltcGxlbWVudGUgdGVuZHLDrWFtb3MgZWwgbnVtZXJvIGRlIGluZGl2aWR1b3MgbXVlc3RyYWwgZGUgbGEgRVBILCBwb3IgZW5kZSBkZWJlbW9zICoqc3VtYXIgbG9zIHZhbG9yZXMgZGUgbGEgdmFyaWFibGUgUE9OREVSQSoqLCBwYXJhIGNvbnRlbXBsYXIgYSBjdWFudGFzIHBlcnNvbmFzIHJlcHJlc2VudGEgY2FkYSBpbmRpdmlkdW8gZW5jdWVzdGFkby4gDQogLSBPY3VwYWRvczogRW4gZXN0ZSBjYXNvLCBkZWJlbW9zIGFncmVnYXIgdW4gKipmaWx0cm8qKiBhbCBwcm9jZWRpbWllbnRvIGFudGVyaW9yLCB5YSBxdWUgdW5pY2FtZW50ZSBxdWVyZW1vcyBzdW1hciBsb3MgcG9uZGVyYWRvcmVzIGRlIGFxdWVsbGFzIHBlcnNvbmFzIHF1ZSBzZSBlbmN1ZW50cmFuIG9jdXBhZGFzLiAoTGEgbMOzZ2ljYSBzZXJpYTogIlN1bWEgbG9zIHZhbG9yZXMgZGUgbGEgY29sdW1uYSBQT05ERVJBLCBzb2xvIHBhcmEgYXF1ZWxsb3MgcmVnaXN0cm9zIGRvbmRlIGVsIEVTVEFETyA9PSAxIikgICAgDQogDQpgYGB7cn0NClBvYmxhY2lvbl9vY3VwYWRvcyA8LSBJbmRpdmlkdWFsX3QxMTcgJT4lIA0KICBzdW1tYXJpc2UoUG9ibGFjaW9uICAgICAgICAgPSBzdW0oUE9OREVSQSksDQogICAgICAgICAgICBPY3VwYWRvcyAgICAgICAgICA9IHN1bShQT05ERVJBW0VTVEFETyA9PSAxXSkpDQoNClBvYmxhY2lvbl9vY3VwYWRvcw0KYGBgDQpMYSBmdW5jacOzbiAgYGBgIHN1bW1hcmlzZSgpIGBgYCBub3MgcGVybWl0ZSBjcmVhciBtdWx0aXBsZXMgdmFyaWFibGVzIGRlIHJlc3VtZW4gYWwgbWlzbW8gdGllbXBvLCAgc2ltcGxlbWVudGUgc2VwYXJhbmRvIGNvbiB1bmEgYGBgICwgYGBgIGNhZGEgdW5vIGRlIGVsbGFzLiBBIHN1IHZleiwgc2UgcHVlZGVuIGNyZWFyIHZhcmlhYmxlcywgYSBwYXJ0aXIgZGUgbGFzIHZhcmlhYmxlcyBjcmVhZGFzIHBvciBsYSBwcm9waWEgZnVuY2nDs24uIERlIGVzdGEgZm9ybWEsIHBvZGVtb3MsIGRpcmVjdGFtZW50ZSBjYWxjdWxhciBsYSAqKnRhc2EgZGUgZW1wbGVvKiogYSBwYXJ0aXIgZGVsIHRvdGFsIHBvYmxhY2lvbmFsIHkgZGUgb2N1cGFkb3MuICAgDQpgYGB7cn0NCkVtcGxlbyA8LSBJbmRpdmlkdWFsX3QxMTcgJT4lIA0KICBzdW1tYXJpc2UoUG9ibGFjaW9uICAgICAgICAgPSBzdW0oUE9OREVSQSksDQogICAgICAgICAgICBPY3VwYWRvcyAgICAgICAgICA9IHN1bShQT05ERVJBW0VTVEFETyA9PSAxXSksDQogICAgICAgICAgICBUYXNhX0VtcGxlbyAgICA9IE9jdXBhZG9zL1BvYmxhY2lvbikNCg0KRW1wbGVvDQpgYGANCg0KRW4gY2FzbyBkZSBxdWVyZXIgZXhwcmVzYXIgbG9zIHJlc3VsdGFkb3MgY29tbyBwb3JjZW50YWplcywgdXRpbGl6YW1vcyBsYSBmdW5jacOzbiBfX3NwcmludGZfXy4gUGFyYSBlbGxvIGRlYmVtb3MgdXRpbGl6YXIgX19tdXRhdGVfXyBwYXJhIHRyYW5zZm9ybWFyIGxhIGNvbHVtbmEgVGFzYV9FbXBsZW8NCmBgYHtyfQ0KRW1wbGVvICU+JSANCiAgbXV0YXRlKFRhc2FfRW1wbGVvX1BvcmMgPSBzcHJpbnRmKCIlMS4xZiUlIiwgMTAwKlRhc2FfRW1wbGVvKSkNCmBgYA0KTsOzdGVzZSBxdWUgZW4gZXN0ZSBjYXNvLCBwYXJhIHBvZGVyIGHDsWFkaXIgZWwgJSwgbGEgZnVuY2nDs24gdHJhbnNmb3JtYSBhIGxhIHZhcmlhYmxlIGVuIHVuIENoYXJhY3RlciwgcG9yIGVuZGUgZGViZSB0ZW5lcnNlIGVuIGN1ZW50YSBxdWUgc2UgcGllcmRlIGxhIGluZm9ybWFjacOzbiBkZWwgbnVtZXJvIGNvbXBsZXRvLg0KDQoNCiMgRXhwb3J0YXIgcmVzdWx0YWRvcyBhICBFeGNlbA0KQ29tbyB2aWVyYW1vcyBlbiBsYSBjbGFzZSAxLCBsYSBmdW5jacOzbiBfX3dyaXRlLnhsc3hfXyBkZSBsYSBsaWJyZXJpYSBfb3Blbnhsc3hfIG5vcyBwZXJtaXRlIGV4cG9ydGFyIGxvcyBkb3MgZGF0YWZyYW1lcyBhIHVuIG1pc21vIGFyY2hpdm8uIENhYmUgYWNsYXJhciBxdWUgZXhpc3RlbiBudW1lcm9zYXMgZnVuY2lvbmVzIHkgbGlicmVyw61hcyBhbHRlcm5hdGl2YXMgcGFyYSBleHBvcnRhciByZXN1bHRhZG9zIGEgdW4gZXhjZWwuIEVuIGVzdGUgY2Fzbywgb3B0YW1vcyBwb3IgX29wZW54bHN4XyB5YSBxdWUgcmVzdWx0YSB1bmEgZGUgbGFzIG3DoXMgc2VuY2lsbGFzIHBhcmEgZXhwb3J0YXIgcmFwaWRhbWVudGUgbG9zIHJlc3VsdGFkb3MuIE90cmFzIGxpYnJlcsOtYXMgcGVybWl0ZW4gdGFtYmnDqW4gZGFyIGZvcm1hdG8gYSBsYXMgdGFibGFzIHF1ZSBzZSBleHBvcnRhbiwgZGVmaW5pciBzaSBkZXNlYW1vcyBzb2JyZWVzY3JpYmlyIGFyY2hpdm9zIGVuIGNhc28gZGUgcXVlIHlhIGV4aXN0YW4sIGV0Yy4gDQoNCmBgYHtyIGV2YWw9RkFMU0UsIHdhcmluaW5nID0gRkFMU0V9DQpMaXN0YV9hX2V4cG9ydGFyIDwtIGxpc3QoIlRhc2EgZGUgRW1wbGVvIiA9IEVtcGxlbywNCiAgICAgICAgICAgICAgICAgICAgICAgICAiUG9ibGFjaW9uIEFnbG9zIiA9IFBvYmxhY2lvbl9BZ2xvbWVyYWRvcykNCg0Kd3JpdGUueGxzeChMaXN0YV9hX2V4cG9ydGFyLCIuLi9SZXN1bHRhZG9zL0luZm9ybWUgTWVyY2FkbyBkZSBUcmFiYWpvLnhsc3giKQ0KDQpgYGANCg0K